# 课程名称: EDA 技术综合设计

设计报告名称:设计六串行整数乘法器

班级: 通信 214

姓名: 王峤宇

学号: 214022

# 一、设计内容及原理

#### 基础任务

设计任务: 输入两个无符号四位二进制(BCD码)整数,求出它们的乘法结果,输入由拨码开关给入,输入输出由数码管十进制显示。

二进制的整数乘法可以通过移位累加的方式实现。二进制整数乘法实现原理如图 1所示, 对于 BCD



图 1: 移位累加过程

码直接相乘,可以直接视为二进制相乘。

### 提高内容

设计任务: 输入两个无符号四位二进制(8421码)整数,求出它们的十进制乘法结果,输入由拨码 开关给入,输入输出由数码管十进制显示。

### 拓展任务

任务要求: 输入两个无符号二进制整数(16-99 范围即可,也可更大但最大 256,这样输出用 5 位数码管,输入用 3 位数码管,刚好够用),求出它们的十进制乘法结果,输入由拨码开关给入,用数码管显示输入输出数值。

设计得到的用于测试整数乘法器的系统模块框图 2所示。



图 2: 系统模块框图

## 二、设计过程

# 基础任务

对于 BCD 码直接相乘的模块设计源文件如下:

Listing 1: 4 位 BCD 码相乘

```
/* 四位bcd乘法器 */
    module mult_bcd(
2
                               /* 时钟 */
3
        input clk,
       input rst,
                               /* 异步高电平复位 */
4
5
       input data_in_rdy,
                               /* 输入数据有效 */
       input [3:0] bcd_in_1, /* 输入乘数 */
        input [3:0] bcd_in_2, /* 输入被乘数 */
        output data_out_rdy, /* 输出数据有效 */
8
        output [3:0] bcd_out_1, /* 输出结果高位 */
9
10
        output [3:0] bcd_out_2 /* 输出结果低位 */
11
       );
12
        reg [1:0] cnt;
13
        always @(posedge clk or posedge rst) begin
           if (rst) begin
14
               cnt <= 2'b0;
15
16
17
           else if (data_in_rdy) begin
18
               cnt <= cnt + 1'b1;
19
           end
20
           else begin
21
             cnt <= 2'b0;
22
23
        end
24
        reg [7:0] num1;
25
       reg [3:0] num2;
26
27
        reg [7:0] acc;
        /* 累加求解乘法 */
28
        always @(posedge clk or posedge rst) begin
29
           if (rst) begin
30
31
               acc <= 8'b0;
32
               num1 <= 8'b0;
33
               num2 <= 4'b0;
34
           else if (data_in_rdy && cnt == 2'b0) begin
35
               num1 <= bcd_in_1 << 1;
36
37
               num2 <= bcd_in_2 >> 1;
38
               acc <= bcd_in_2[0] ? bcd_in_1 : 8'b0;
39
           \verb"end"
           else if (cnt != 2'd3) begin
40
               num1 <= num1 << 1;
41
42
               num2 <= num2 >> 1;
               acc <= num2[0] ? num1 + acc : acc;
44
```

```
45
             else begin
46
                 acc <= acc;</pre>
47
             end
48
         end
49
50
        reg [7:0] res;
51
         reg data_out_rdy_r;
         always @(posedge clk or posedge rst) begin
52
53
             if (rst) begin
54
                 data_out_rdy_r <= 1'b0;</pre>
55
                 res <= 8'b0;
56
             else if (cnt == 2'd3) begin
57
58
                 data_out_rdy_r <= 1'b1;</pre>
59
                 res <= acc;
60
             end
61
             else begin
62
                 data_out_rdy_r <= 1'b0;</pre>
                 res <= 8'b0;
63
64
             end
65
         end
66
         assign data_out_rdy = data_out_rdy_r;
67
         wire [3:0] bcd_foobar;
68
         bin2bcd_8bit bin2bcd_8bit_inst (
69
70
             .bin(res),
71
             .bcd_12bit({bcd_foobar, bcd_out_1, bcd_out_2})
72
        );
73
    endmodule
```

其中二进制转 BCD 码源文件如下:

Listing 2: 二进制转 BCD 码

```
module bin2bcd_8bit(
1
2
        input [7:0] bin,
3
        output [11:0] bcd_12bit /* 高中低BCD结果 */
4
       );
5
6
       integer i;
7
       reg [19:0] bcd_temp;
        always @(*) begin
           // 初始化, 直接移位三次
9
10
           bcd_temp = {9'b0, bin, 3'b0};
           // 逐位移位法
11
           for (i = 0; i < 5; i = i + 1'b1) begin</pre>
12
               // 如果BCD的每一部分 > 4, 加3
13
14
                if (bcd_temp[11:8] > 4)
                    bcd_temp[11:8] = bcd_temp[11:8] + 3;
15
               if (bcd_temp[15:12] > 4)
16
17
                   bcd_temp[15:12] = bcd_temp[15:12] + 3;
               if (bcd_temp[19:16] > 4)
18
19
                    bcd_temp[19:16] = bcd_temp[19:16] + 3;
```

top 文件如下:

Listing 3: top

```
module mult_bcd_top(
1
2
        input clk,
                                       /* 时钟 */
                                       /* 异步高电平复位 */
3
        input rst,
        input [7:0] keys,
                                       /* 8个拨码开关输入 */
        input [7:0] dip,
5
                                       /* dip开关输入 */
        output [7:0] sseg1, sseg2,
                                       /* 八段数码管 */
6
7
        output [3:0] an1, an2
                                       /* 数码管片选信号 */
8
       );
9
        /* 例化时分复用的乘法器模块 */
10
        wire [3:0] bcd_out_1;
11
        wire [3:0] bcd_out_2;
12
13
       wire data_out_rdy;
        mult_bcd mult_bcd_inst (
14
15
           .clk(clk),
16
           .rst(rst),
            .data_in_rdy(1'b1),
17
            .bcd_in_1(keys[3:0]),
18
19
            .bcd_in_2(dip[3:0]),
20
            .data_out_rdy(data_out_rdy),
21
            .bcd_out_1(bcd_out_1),
            .bcd_out_2(bcd_out_2)
22
23
       );
        /* 第一个数码管显示 */
24
25
        reg [3:0] dsp_buffer_1[0:3];
26
        always @(posedge clk) begin
            dsp_buffer_1[0] <= keys[3:0];
27
           dsp_buffer_1[1] <= 4'd10;
28
           dsp_buffer_1[2] <= 4'd10;
29
           dsp_buffer_1[3] <= dip[3:0];
30
31
        /* 第二个数码管显示 */
32
33
        reg [3:0] dsp_buffer_2[0:3];
        always @(posedge clk or posedge rst) begin
34
35
            if (rst) begin
               dsp_buffer_2[0] <= 4'd10;
36
37
                dsp_buffer_2[1] <= 4'd10;
                dsp_buffer_2[2] <= 4'd10;
38
                dsp_buffer_2[3] <= 4'd10;
39
40
           end
41
            else if (data_out_rdy) begin
42
                dsp_buffer_2[0] <= 4'd10;
```

```
43
                dsp_buffer_2[1] <= 4'd10;
                dsp_buffer_2[2] <= bcd_out_1;
44
                dsp_buffer_2[3] <= bcd_out_2;</pre>
45
46
            end
47
            else begin
48
                dsp_buffer_2[0] <= 4'd10;
                dsp_buffer_2[1] <= 4'd10;
49
                dsp_buffer_2[2] <= dsp_buffer_2[2];</pre>
50
                dsp_buffer_2[3] <= dsp_buffer_2[3];</pre>
51
52
            end
53
        /* 第一个四位数码管 */
54
        scan_led_hex_disp_4 scan_led_hex_disp_4_inst_1 (
55
56
            .clk(clk), .rst(rst),
57
            .hex0(dsp_buffer_1[0]), .hex1(dsp_buffer_1[1]),
58
            .hex2(dsp_buffer_1[2]), .hex3(dsp_buffer_1[3]),
59
            .dp(4'b0000), .an(an1), .sseg(sseg1)
60
        );
        /* 第二个四位数码管 */
61
62
        scan_led_hex_disp_4 scan_led_hex_disp_4_inst_2 (
63
            .clk(clk), .rst(rst),
64
            .hex0(dsp_buffer_2[0]), .hex1(dsp_buffer_2[1]),
            .hex2(dsp_buffer_2[2]), .hex3(dsp_buffer_2[3]),
65
            .dp(4'b0000), .an(an2), .sseg(sseg2)
66
67
        );
68
    endmodule
```

针对模块的仿真文件如下:

Listing 4: BCD 乘法器仿真文件

```
module mult_bcd_tb;
2
        //Ports
3
        reg clk;
4
        reg rst;
5
        reg data_in_rdy;
 6
        reg [3:0] bcd_in_1;
7
        reg [3:0] bcd_in_2;
8
        wire data_out_rdy;
        wire [3:0] bcd_out_1;
9
        wire [3:0] bcd_out_2;
10
11
12
        reg [3:0] bcd_res[0:1];
13
        initial begin
            clk = 0;
14
15
            rst = 0;
16
            #10;
17
            rst = 1;
            #10;
18
            rst = 0;
19
20
            bcd_in_1 = 4'd4;
            bcd_in_2 = 4'd5;
21
22
            data_in_rdy = 1'b1;
```

```
23
24
            wait(data_out_rdy == 1'b1);
            data_in_rdy = 1'b0;
25
26
            bcd_res[0] = bcd_out_1;
27
            bcd_res[1] = bcd_out_2;
28
29
            $finish;
        end
30
31
32
        mult_bcd mult_bcd_inst (
33
            .clk(clk),
34
            .rst(rst),
35
            .data_in_rdy(data_in_rdy),
36
            .bcd_in_1(bcd_in_1),
            .bcd_in_2(bcd_in_2),
37
38
            .data_out_rdy(data_out_rdy),
39
            .bcd_out_1(bcd_out_1),
40
             .bcd_out_2(bcd_out_2)
41
        );
42
43
        always #5 clk = ! clk ;
44
    endmodule
```

基础任务、调高任务以及拓展任务公用的约束文件如下:

Listing 5: 约束文件

```
/* 100MHz 时钟信号 */
 1
    set_property -dict {PACKAGE_PIN P17 IOSTANDARD LVCMOS33} [get_ports clk]
2
   /* 复位信号 */
3
    set_property -dict {PACKAGE_PIN R11 IOSTANDARD LVCMOS33} [get_ports rst]
4
    /* 拨码开关 */
6
    set_property -dict {PACKAGE_PIN P5 IOSTANDARD LVCMOS33} [get_ports {keys[7]}]
    set_property -dict {PACKAGE_PIN P4 IOSTANDARD LVCMOS33} [get_ports {keys[6]}]
7
   set_property -dict {PACKAGE_PIN P3 IOSTANDARD LVCMOS33} [get_ports {keys[5]}]
8
9
    set_property -dict {PACKAGE_PIN P2 IOSTANDARD LVCMOS33} [get_ports {keys[4]}]
    set_property -dict {PACKAGE_PIN R2 IOSTANDARD LVCMOS33} [get_ports {keys[3]}]
10
11
    set_property -dict {PACKAGE_PIN M4 IOSTANDARD LVCMOS33} [get_ports {keys[2]}]
    set_property -dict {PACKAGE_PIN N4 IOSTANDARD LVCMOS33} [get_ports {keys[1]}]
12
   set_property -dict {PACKAGE_PIN R1 IOSTANDARD LVCMOS33} [get_ports {keys[0]}]
13
14
    /* DIP */
    set_property -dict {PACKAGE_PIN U3 IOSTANDARD LVCMOS33} [get_ports {dip[7]}]
15
16
    set_property -dict {PACKAGE_PIN U2 IOSTANDARD LVCMOS33} [get_ports {dip[6]}]
    set_property -dict {PACKAGE_PIN V2 IOSTANDARD LVCMOS33} [get_ports {dip[5]}]
17
    set_property -dict {PACKAGE_PIN V5 IOSTANDARD LVCMOS33} [get_ports {dip[4]}]
18
19
    set_property -dict {PACKAGE_PIN V4 IOSTANDARD LVCMOS33} [get_ports {dip[3]}]
    set_property -dict {PACKAGE_PIN R3 IOSTANDARD LVCMOS33} [get_ports {dip[2]}]
20
    set_property -dict {PACKAGE_PIN T3 IOSTANDARD LVCMOS33} [get_ports {dip[1]}]
21
    set_property -dict {PACKAGE_PIN T5 IOSTANDARD LVCMOS33} [get_ports {dip[0]}]
22
   /* 数码管 */
23
24
   set_property -dict {PACKAGE_PIN G2 IOSTANDARD LVCMOS33} [get_ports {an1[0]}]
    set_property -dict {PACKAGE_PIN C2 IOSTANDARD LVCMOS33} [get_ports {an1[1]}]
    set_property -dict {PACKAGE_PIN C1 IOSTANDARD LVCMOS33} [get_ports {an1[2]}]
```

```
27
    set_property -dict {PACKAGE_PIN H1 IOSTANDARD LVCMOS33} [get_ports {an1[3]}]
    set_property -dict {PACKAGE_PIN G6 IOSTANDARD LVCMOS33} [get_ports {an2[3]}]
28
    set_property -dict {PACKAGE_PIN E1 IOSTANDARD LVCMOS33} [get_ports {an2[2]}]
29
    set_property -dict {PACKAGE_PIN F1 IOSTANDARD LVCMOS33} [get_ports {an2[1]}]
30
    set_property -dict {PACKAGE_PIN G1 IOSTANDARD LVCMOS33} [get_ports {an2[0]}]
31
32
    set_property -dict {PACKAGE_PIN D5 IOSTANDARD LVCMOS33} [get_ports {sseg1[7]}]
    set_property -dict {PACKAGE_PIN B4 IOSTANDARD LVCMOS33} [get_ports {sseg1[6]}]
33
    set_property -dict {PACKAGE_PIN A4 IOSTANDARD LVCMOS33} [get_ports {sseg1[5]}]
34
    set_property -dict {PACKAGE_PIN A3 IOSTANDARD LVCMOS33} [get_ports {sseg1[4]}]
35
    set_property -dict {PACKAGE_PIN B1 IOSTANDARD LVCMOS33} [get_ports {sseg1[3]}]
36
37
    set_property -dict {PACKAGE_PIN A1 IOSTANDARD LVCMOS33} [get_ports {sseg1[2]}]
    set_property -dict {PACKAGE_PIN B3 IOSTANDARD LVCMOS33} [get_ports {sseg1[1]}]
38
    set_property -dict {PACKAGE_PIN B2 IOSTANDARD LVCMOS33} [get_ports {sseg1[0]}]
39
    set_property -dict {PACKAGE_PIN H2 IOSTANDARD LVCMOS33} [get_ports {sseg2[7]}]
40
    set_property -dict {PACKAGE_PIN D4 IOSTANDARD LVCMOS33} [get_ports {sseg2[6]}]
41
42
    set_property -dict {PACKAGE_PIN E3 IOSTANDARD LVCMOS33} [get_ports {sseg2[5]}]
    set_property -dict {PACKAGE_PIN D3 IOSTANDARD LVCMOS33} [get_ports {sseg2[4]}]
43
    set_property -dict {PACKAGE_PIN F4 IOSTANDARD LVCMOS33} [get_ports {sseg2[3]}]
44
    set_property -dict {PACKAGE_PIN F3 IOSTANDARD LVCMOS33} [get_ports {sseg2[2]}]
45
    set_property -dict {PACKAGE_PIN E2 IOSTANDARD LVCMOS33} [get_ports {sseg2[1]}]
46
    set_property -dict {PACKAGE_PIN D2 IOSTANDARD LVCMOS33} [get_ports {sseg2[0]}]
```

## 提高任务

四位二进制乘法源码如下

Listing 6: 四位二进制乘法源码

```
1
    module mult_bin(
2
        input clk,
                               /* 时钟 */
                               /* 异步高电平复位 */
3
        input rst,
        input data_in_rdy,
                               /* 输入数据有效 */
4
        input [7:0] data_in_1, /* 输入乘数 */
5
        input [7:0] data_in_2, /* 输入被乘数 */
6
                               /* 输出数据有效 */
7
        output data_out_rdy,
        output [15:0] data_out /* 输出8位数据 */
8
9
10
        reg [2:0] cnt;
11
        always @(posedge clk or posedge rst) begin
12
            if (rst) begin
                cnt <= 2'b0;
13
14
15
            else if (data_in_rdy) begin
                cnt <= cnt + 1'b1;</pre>
16
17
           end
18
            else begin
19
               cnt <= 2'b0:
20
           end
21
22
       reg [15:0] num1;
23
```

```
24
        reg [7:0] num2;
        reg [15:0] acc;
25
        /* 累加求解乘法 */
26
27
        always @(posedge clk or posedge rst) begin
28
            if (rst) begin
29
                 acc <= 16'b0;
                 num1 <= 16'b0;
30
                 num2 <= 8'b0;
31
32
            end
            else if (data_in_rdy && cnt == 2'b0) begin
33
34
                 num1 <= data_in_1 << 1;</pre>
35
                 num2 <= data_in_2 >> 1;
                 acc <= data_in_2[0] ? data_in_1 : 16'b0;</pre>
36
37
            end
38
            else if (cnt != 3'd7) begin
39
                 num1 <= num1 << 1;
40
                 num2 <= num2 >> 1;
                 acc <= num2[0] ? num1 + acc : acc;
41
42
            end
43
            else begin
44
                acc <= acc;
45
46
        end
47
        reg [15:0] res;
48
49
        reg data_out_rdy_r;
50
        always @(posedge clk or posedge rst) begin
             if (rst) begin
51
52
                 data_out_rdy_r <= 1'b0;</pre>
                 res <= 16'b0;
53
54
            end
55
            else if (cnt == 3'd7) begin
56
                 data_out_rdy_r <= 1'b1;</pre>
57
                res <= acc;
58
            end
59
            else begin
60
                data_out_rdy_r <= 1'b0;</pre>
61
                 res <= 16'b0;
62
            end
63
64
        assign data_out_rdy = data_out_rdy_r;
        assign data_out = res;
65
    endmodule
```

top 文件如下:

Listing 7: top

```
6
        output [7:0] sseg1, sseg2,
                                     /* 八段数码管 */
7
        output [3:0] an1, an2
                                         /* 数码管片选信号 */
8
        );
9
        /* 例化时分复用的乘法器模块 */
10
11
        wire [15:0] res;
12
        wire data_out_rdy;
        mult_bin mult_bin_inst (
13
14
            .clk(clk),
15
            .rst(rst),
16
            .data_in_rdy(1'b1),
17
            .data_in_1(keys),
            .data_in_2(dip),
18
19
            .data_out_rdy(data_out_rdy),
20
            .data_out(res)
21
        );
22
        /* 第一个数码管显示 */
23
        wire [3:0] dsp_buffer_1[0:3];
24
25
        bin2bcd_8bit bin2bcd_8bit_inst_1 (
26
            .bin(keys),
27
            .bcd_12bit({dsp_buffer_1[0], dsp_buffer_1[1]})
28
        );
29
        bin2bcd_8bit bin2bcd_8bit_inst_2 (
            .bin(dip),
30
31
            .bcd_12bit({dsp_buffer_1[2], dsp_buffer_1[3]})
32
        );
33
        /* 第二个数码管显示 */
34
        wire [3:0] dsp_buffer_2[0:3];
35
36
        bin2bcd_8bit bin2bcd_8bit_inst (
37
            .bin(res),
            .bcd_12bit({dsp_buffer_2[1], dsp_buffer_2[2], dsp_buffer_2[3]})
38
39
        );
40
41
        reg [3:0] dsp_buffer_2_r[0:3];
        always @(posedge clk or posedge rst) begin
42
43
            if (rst) begin
44
                dsp_buffer_2_r[0] <= 4'd0;
                dsp_buffer_2_r[1] <= 4'd0;
45
46
                dsp_buffer_2_r[2] <= 4'd0;
                dsp_buffer_2_r[3] <= 4'd0;
47
48
            else if (data_out_rdy == 1'b1) begin
49
                dsp_buffer_2_r[0] <= dsp_buffer_2[0];</pre>
50
51
                dsp_buffer_2_r[1] <= dsp_buffer_2[1];</pre>
52
                dsp_buffer_2_r[2] <= dsp_buffer_2[2];</pre>
53
                dsp_buffer_2_r[3] <= dsp_buffer_2[3];</pre>
54
            end
55
            else begin
56
                dsp_buffer_2_r[0] <= dsp_buffer_2_r[0];</pre>
57
                dsp_buffer_2_r[1] <= dsp_buffer_2_r[1];</pre>
```

```
58
                dsp_buffer_2_r[2] <= dsp_buffer_2_r[2];</pre>
                dsp_buffer_2_r[3] <= dsp_buffer_2_r[3];</pre>
59
60
            end
61
        end
62
63
        /* 第一个四位数码管 */
64
        scan_led_hex_disp_4 scan_led_hex_disp_4_inst_1 (
            .clk(clk), .rst(rst),
65
            .hex0(dsp_buffer_1[0]), .hex1(dsp_buffer_1[1]),
66
            .hex2(dsp_buffer_1[2]), .hex3(dsp_buffer_1[3]),
67
68
            .dp(4'b0000), .an(an1), .sseg(sseg1)
69
        );
        /* 第二个四位数码管 */
70
71
        scan_led_hex_disp_4 scan_led_hex_disp_4_inst_2 (
72
            .clk(clk), .rst(rst),
73
            .hex0(dsp_buffer_2[0]), .hex1(dsp_buffer_2[1]),
74
            .hex2(dsp_buffer_2[2]), .hex3(dsp_buffer_2[3]),
            .dp(4'b0000), .an(an2), .sseg(sseg2)
75
76
        );
77
    endmodule
```

仿真文件如下:

Listing 8: 仿真文件

```
1
    module mult_bin_tb;
2
        reg clk;
3
        reg rst;
4
        reg data_in_rdy;
        reg [7:0] data_in_1;
5
6
        reg [7:0] data_in_2;
7
        wire data_out_rdy;
        wire [15:0] data_out;
8
9
10
        initial begin
11
            rst = 0;
12
            clk = 0;
13
            #10;
14
            rst = 1;
            #10;
15
            rst = 0;
16
17
            data_in_rdy = 1'b1;
18
            data_in_1 = 8'd15;
19
            data_in_2 = 8'd7;
20
            wait(data_out_rdy == 1'b1);
21
            #100;
22
            $finish;
23
        end
24
        mult_bin mult_bin_inst (
25
            .clk(clk),
26
27
            .rst(rst),
28
            .data_in_rdy(data_in_rdy),
```

### 拓展任务

源文件:

Listing 9: 8 位乘法器的源文件

```
1
    module mult_bin(
2
        input clk,
                               /* 时钟 */
                                /* 异步高电平复位 */
3
        input rst,
                               /* 输入数据有效 */
        input data_in_rdy,
4
        input [7:0] data_in_1, /* 输入乘数 */
5
       input [7:0] data_in_2, /* 输入被乘数 */
6
        output data_out_rdy,
                               /* 输出数据有效 */
        output [15:0] data_out /* 输出8位数据 */
8
9
       );
10
        reg [2:0] cnt;
11
        always @(posedge clk or posedge rst) begin
12
            if (rst) begin
13
                cnt <= 2'b0;
14
15
            else if (data_in_rdy) begin
16
               cnt <= cnt + 1'b1;
17
           end
18
            else begin
                cnt <= 2'b0;
19
20
            end
21
        end
22
23
        reg [15:0] num1;
        reg [7:0] num2;
24
        reg [15:0] acc;
25
        /* 累加求解乘法 */
26
27
        always @(posedge clk or posedge rst) begin
28
           if (rst) begin
29
                acc <= 16'b0;
30
                num1 <= 16'b0;
               num2 <= 8'b0;
31
32
            else if (data_in_rdy && cnt == 2'b0) begin
33
34
                num1 <= data_in_1 << 1;</pre>
35
                num2 <= data_in_2 >> 1;
                acc <= data_in_2[0] ? data_in_1 : 16'b0;</pre>
36
```

```
37
             else if (cnt != 3'd7) begin
38
                 num1 <= num1 << 1;
39
40
                 num2 <= num2 >> 1;
                 acc <= num2[0] ? num1 + acc : acc;
41
42
43
             else begin
                 acc <= acc;
44
45
             end
46
        end
47
48
        reg [15:0] res;
        reg data_out_rdy_r;
49
50
        always @(posedge clk or posedge rst) begin
51
             if (rst) begin
52
                 data_out_rdy_r <= 1'b0;</pre>
53
                 res <= 16'b0;
54
55
             else if (cnt == 3'd7) begin
                 data_out_rdy_r <= 1'b1;</pre>
56
57
                 res <= acc;
58
59
             else begin
60
                 data_out_rdy_r <= 1'b0;</pre>
                 res <= 16'b0;
61
62
             end
63
64
        assign data_out_rdy = data_out_rdy_r;
65
        assign data_out = res;
    endmodule
66
```

16 位二进制转 BCD 码模块源文件:

Listing 10: 16bit 二进制转 BCD 码

```
1
    module bin2bcd_16bit(
2
       input [15:0] bin,
        output [19:0] bcd_20bit /* 高中低BCD结果 */
3
4
       );
5
6
        integer i;
7
       reg [35:0] bcd_temp;
8
        always @(*) begin
           // 初始化, 直接移位三次
9
10
           bcd_temp = {17'b0, bin, 3'b0};
           // 逐位移位法
11
12
           for (i = 0; i < 13; i = i + 1'b1) begin</pre>
                // 如果BCD的每一部分 > 4, 加3
13
               if (bcd_temp[19:16] > 4)
14
                   bcd_temp[19:16] = bcd_temp[19:16] + 3;
15
16
               if (bcd_temp[23:20] > 4)
                   bcd_temp[23:20] = bcd_temp[23:20] + 3;
17
18
                if (bcd_temp[27:24] > 4)
```

```
19
                     bcd_temp[27:24] = bcd_temp[27:24] + 3;
                 if (bcd_temp[31:28] > 4)
20
                     bcd_temp[31:28] = bcd_temp[31:28] + 3;
21
                 if (bcd_temp[35:32] > 4)
22
                     bcd_temp[35:32] = bcd_temp[35:32] + 3;
23
24
                 // 左移1位
25
                 bcd_temp = {bcd_temp[34:0], 1'b0};
26
            \verb"end"
27
        end
28
        assign bcd_20bit = bcd_temp[35-:20];
29
```

top 文件:

Listing 11: 8 位乘法器 top 文件

```
module mult_bin_top(
1
2
        input clk,
                                       /* 时钟 */
                                       /* 异步高电平复位 */
3
        input rst,
        input [7:0] keys,
                                       /* 8个拨码开关输入 */
4
        input [7:0] dip,
                                       /* dip开关输入 */
                                      /* 八段数码管 */
        output [7:0] sseg1, sseg2,
6
7
        output [3:0] an1, an2
                                      /* 数码管片选信号 */
8
       );
9
10
        /* 例化时分复用的乘法器模块 */
11
        wire [15:0] res;
        wire data_out_rdy;
12
13
       mult_bin mult_bin_inst (
           .clk(clk),
14
15
           .rst(rst),
16
           .data_in_rdy(1'b1),
17
            .data_in_1(keys),
18
            .data_in_2(dip),
            .data_out_rdy(data_out_rdy),
19
20
            .data_out(res)
21
       );
22
        /* 第一个数码管显示 */
23
        wire [3:0] dsp_buffer_1[0:3];
24
25
        bin2bcd_8bit bin2bcd_8bit_inst_1 (
26
            .bin(keys),
27
            .bcd_12bit({dsp_buffer_1[0], dsp_buffer_1[1]})
28
       );
        bin2bcd_8bit bin2bcd_8bit_inst_2 (
29
30
            .bin(dip),
            .bcd_12bit({dsp_buffer_1[2], dsp_buffer_1[3]})
31
32
33
        /* 第二个数码管显示 */
34
35
        wire [3:0] dsp_buffer_2[0:3];
        bin2bcd_16bit bin2bcd_16bit_inst (
36
37
            .bin(res),
```

```
38
             .bcd_20bit({dsp_buffer_2[0], dsp_buffer_2[1], dsp_buffer_2[2], dsp_buffer_2[3]})
39
        );
40
41
        reg [3:0] dsp_buffer_2_r[0:3];
42
        always @(posedge clk or posedge rst) begin
43
             if (rst) begin
44
                 dsp_buffer_2_r[0] <= 4'd0;
                 dsp_buffer_2_r[1] <= 4'd0;
45
46
                 dsp_buffer_2_r[2] <= 4'd0;
47
                 dsp_buffer_2_r[3] <= 4'd0;
48
49
             else if (data_out_rdy == 1'b1) begin
                 dsp_buffer_2_r[0] <= dsp_buffer_2[0];</pre>
50
51
                 dsp_buffer_2_r[1] <= dsp_buffer_2[1];</pre>
52
                 dsp_buffer_2_r[2] <= dsp_buffer_2[2];</pre>
                 dsp_buffer_2_r[3] <= dsp_buffer_2[3];</pre>
53
54
55
             else begin
                 dsp_buffer_2_r[0] <= dsp_buffer_2_r[0];</pre>
56
                 dsp_buffer_2_r[1] <= dsp_buffer_2_r[1];</pre>
57
58
                 dsp_buffer_2_r[2] <= dsp_buffer_2_r[2];</pre>
59
                 dsp_buffer_2_r[3] <= dsp_buffer_2_r[3];</pre>
60
             end
61
        end
62
63
         /* 第一个四位数码管 */
        scan_led_hex_disp_4 scan_led_hex_disp_4_inst_1 (
64
65
             .clk(clk), .rst(rst),
66
             .hex0(dsp_buffer_1[0]), .hex1(dsp_buffer_1[1]),
             .hex2(dsp_buffer_1[2]), .hex3(dsp_buffer_1[3]),
67
68
             .dp(4'b0000), .an(an1), .sseg(sseg1)
69
        );
        /* 第二个四位数码管 */
70
71
        \verb|scan_led_hex_disp_4| & \verb|scan_led_hex_disp_4_inst_2| \\
             .clk(clk), .rst(rst),
72
73
             .hex0(dsp_buffer_2[0]), .hex1(dsp_buffer_2[1]),
             .hex2(dsp_buffer_2[2]), .hex3(dsp_buffer_2[3]),
74
75
             .dp(4'b0000), .an(an2), .sseg(sseg2)
76
        );
77
    endmodule
```

仿真文件:

Listing 12: 8 位乘法器仿真文件

```
module mult_bin_tb;
1
2
       reg clk;
3
       reg rst;
4
       reg data_in_rdy;
       reg [7:0] data_in_1;
5
6
       reg [7:0] data_in_2;
7
       wire data_out_rdy;
8
       wire [15:0] data_out;
```

```
9
10
        initial begin
            rst = 0;
11
12
            clk = 0;
            #10;
13
14
            rst = 1;
            #10;
15
            rst = 0;
16
17
            data_in_rdy = 1'b1;
            data_in_1 = 8'd77;
18
19
            data_in_2 = 8'd33;
            wait(data_out_rdy == 1'b1);
20
            #100;
21
22
            $finish;
23
        end
24
        mult_bin mult_bin_inst (
25
            .clk(clk),
26
27
            .rst(rst),
            .data_in_rdy(data_in_rdy),
28
29
            .data_in_1(data_in_1),
30
            .data_in_2(data_in_2),
31
            .data_out_rdy(data_out_rdy),
32
            .data_out(data_out)
33
        );
34
35
        always #5 clk = ! clk ;
    endmodule
36
```

top 仿真文件:

Listing 13: top 仿真文件

```
module mult_bin_top_tb;
1
2
3
        // Parameters
4
5
        //Ports
        reg clk;
6
7
        reg rst;
        reg [7:0] keys;
8
9
        reg [7:0] dip;
        wire [7:0] sseg1;
10
        wire [7:0] sseg2;
11
        wire [3:0] an1;
12
        wire [3:0] an2;
13
14
        initial begin
            rst = 0;
15
16
            clk = 0;
            #10;
17
18
            rst = 1;
19
            #10;
20
            rst = 0;
```

```
21
            keys = 8'd15;
22
            dip = 8'd7;
            #100;
23
24
             $finish;
25
        end
26
27
        mult_bin_top mult_bin_top_inst (
            .clk(clk),
28
29
             .rst(rst),
            .keys(keys),
30
31
             .dip(dip),
             .sseg1(sseg1),
32
             .sseg2(sseg2),
33
             .an1(an1),
34
35
             .an2(an2)
36
        );
37
        always #5 clk = ! clk ;
38
39
40
    endmodule
```

# 三、仿真结果

## 基础任务



图 3: bcd 乘法器的仿真结果

针对 BCD 乘法器的仿真结果如图 3所示,可以看到随着输入周期的变化,系统按照时分复用的流程进行计算,当指定输入数据有效之后,经过四个周期完成运算,得到 BCD 码 4 和 5 的乘积结果位 20。

## 提高任务



图 4: 四位二进制乘法的仿真结果

四位二进制乘法的验证结果如图 4所示, 其计算流程与 BCD 乘法器设计的工作流程一致, 计算结果正确无误。

## 拓展任务



图 5: 八位二进制乘法的仿真结果

八位二进制乘法的验证结果如图 5所示, 通过 4 位二进制乘法改变位宽实现,8 个周期后完成计算, 计算结果 77\*33=2541。

# 四、硬件验证结果

# 基础任务



图 6: 5\*5



图 7: 9\*7

硬件验证结果如图 6和图 7所示, 计算结果正确。

## 提高任务



图 8: 14\*6 = 84

四位二进制整数乘法的硬件验证结果如图 8, 计算 14\*6 得到结果 84, 结果无误。

## 拓展任务



图 9: 98\*83 = 8134

8 位二进制整数乘法的硬件验证结果如图 9, 计算 98\*83 得到结果 8134, 结果无误。

## 五、问题解决

#### 加载 bit 文件后, 数码管显示不正确

解决: 仿真时主要是对模块进行仿真, top 文件中完成的显示和输入没有进行直接的仿真, 导致未发现此次数码管问题, 烧录后数码管仅显示第一位, 起初以为是约束文件出错, 之后通过更改输入的方式发现输出一直, 得到结论: 问题出在数码管的部分。

首先通过 RTL 原理图检查问题,可以直接发现是数码管例化时给的时钟错误, Vivado 下的 Verilog 语法检查器和综合器不会对未经声明而直接用于模块例化的信号产生 error, 仅仅会产生 warning, 导致此次错误发生。

### 构建 16bit 的 bin2bcd 模块, 不输出

解决: 图方便直接采用 8bit 的模块修改得到 16bit 的模块, 但是输出部分忘了修改, 新模块的输出是bcd 20bit, 但没有修改原先的语句的左变量:

assign bcd\_12bit = bcd\_temp[35-:20];

但输出的 port 位 bcd\_20bit, 导致错误, 通过查阅资料可以得知, Verilog 中支持未声明的线网类型数据, 如 wire, 直接使用 assign 语句声明的同时为其建立组合逻辑关系。所以综合器没有报错, 最终通过仿真层级关系一层一层检查, 最后发现问题。

#### 修改代码后, 反复重新生成 bit 文件没有效果

修改代码后,首先检查仿真结果,仿真无误,但是板上验证结果不对,推测为 Vivado 在从综合开始的 重新生成 bit 文件的过程中,文件被缓存,更改后的代码没有有效进行综合,重启 vivado 后再次进行 生成 bit 文件,得到正确的结果。

# 六、写出心得体会

本次设计过程中通过 FPGA 完成了简单的算法设计,也了解到 FPGA 中设计复杂算法的几种方法,如果不采用合理的算法和设计结构将导致系统的资源占用过多,甚至无法实现。基本的设计流程思路有并行和串行设计,串行体现的是对资源的时分复用,而并行就是构建好基础模块后,重复模块实现复杂算法,为了减少资源的占用,通常选用串行设计思路。

为了解决串行计算中时分复用导致的效率低下,通常还会采用流水线的设计方式,将复杂的算法分为不同的处理阶段,数据串行输入,同一周期内不同数据进行不同的操作处理,最终实现的效果就是一个周期可以得到一个计算结果,在效率和资源上达到一定的平衡。